package com.atguigu.gulimall.order.config;

import org.springframework.amqp.core.Message;
import org.springframework.amqp.rabbit.connection.CorrelationData;
import org.springframework.amqp.rabbit.core.RabbitTemplate;
import org.springframework.amqp.support.converter.Jackson2JsonMessageConverter;
import org.springframework.amqp.support.converter.MessageConverter;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

import javax.annotation.PostConstruct;

/**
 * @Classname MyRabbitConfig
 * @Description TODO
 * @Date 2024/7/14 15:03
 * @Created by 369502594
 */
@Configuration
public class MyRabbitConfig {

    @Autowired
    RabbitTemplate rabbitTemplate;

    @Bean
    public MessageConverter messageConverter() {
        return new Jackson2JsonMessageConverter();
    }

    /**
     * 定制RabbitTemplate
     * 1、服务器受到消息就回调
     *      1、spring.rabbitmq.publisher-confirms=true
     *      2、设置确认回调 ConfirmCallback
     * 2、消息正确抵达队列进行回调
     *      1、spring.rabbitmq.publisher-returns=true
     *         spring.rabbitmq.template.mandatory=true
     *      2、设置确认回调 ReturnCallback
     * 3、消费端确认（保证每个消息被正确消费，此时才可以broker删除这条消息）
     *      spring.rabbitmq.listener.simple.acknowledge-mode=manual
     *      1、默认是自动确认的，只要消息接收到，客户端会自动确认，服务端就会移除这个消息
     *          1）、问题：
     *              我们收到很多消息，自动回复给服务器ack，只有一个消息处理成功，宕机了，发生消息丢失
     *              手动确认。只要我们没有明确告诉MQ，消息没有被正确处理，消息就回滚到队列，保持uncheck状态。即使Consumer宕机，消息不会丢失，会重新变为Ready状态，
     *              下次有新的Consumer连接进来就发给他
     *          2）、如何签收
     *              channel.basicAck(deliveryTag,false); 签收，业务成功就应该签收
     *              channel.basicNack(deliveryTag,false,true);  拒签，业务失败应该拒签
     */
    @PostConstruct // MyRabbitConfig对象创建完成以后，执行这个方法
    public void initRabbitTemplate() {
        // 设置确认回调
        rabbitTemplate.setConfirmCallback(new RabbitTemplate.ConfirmCallback() {
            /**
             * 1、只要消息抵达Broker就 ack=true
             * @param correlationData correlation data for the callback. 当前消息的唯一关联数据（这个是消息的唯一id）
             * @param ack true for ack, false for nack 消息是否成功受到
             * @param cause An optional cause, for nack, when available, otherwise null.    失败的原因
             */
            @Override
            public void confirm(CorrelationData correlationData, boolean ack, String cause) {
                /**
                 * 1、做好消息确认机制（publish、consumer【手动ack】）
                 * 2、每一个发送的消息都在数据库做好记录。定期将失败的消息再次发送一遍
                 *
                 * // 服务器收到了
                 * // 修改消息的状态
                 */
                // 服务器收到了
                System.out.println("confirm...correlationData[" + correlationData + "]==>ack:[" + ack + "]==>cause:[" + cause + "]");
            }
        });

        // 设置消息抵达队列的确认回调
        rabbitTemplate.setReturnCallback(new RabbitTemplate.ReturnCallback() {

            /**
             * 只要消息没有投递给指定的队列，就触发这个失败回调
             * @param message the returned message. 投递失败的消息详细信息
             * @param replyCode the reply code.     回复的状态码
             * @param replyText the reply text.     回复的文本内容
             * @param exchange the exchange.        当时这个消息发给哪个交换机
             * @param routingKey the routing key.   当时这个消息用哪个路由键
             */
            @Override
            public void returnedMessage(Message message, int replyCode, String replyText, String exchange, String routingKey) {
                // 报错误。修改数据库当前消息的错误状态 -> 错误
                System.out.println("Fail Message [" + message + "], replyCode[" + replyCode + "], replyText[" + replyText + "], exchange[" + exchange + "], routingKey[" + routingKey + "]");
            }
        });
    }
}
